from datetime import datetime
import random
import re

from flask import request, abort, current_app, make_response, jsonify, session

from info import sr, db
from info.lib.yuntongxun.sms import CCP
from info.modules.passport import passport_blu
from info.static.models import User

from info.utils.captcha.pic_captcha import captcha



# 使用蓝图来装饰路由
# 获取图片验证码
from info.utils.response_code import RET, error_map


@passport_blu.route("/get_img_code")
def get_img_code():
    # 获取参数
    img_code_id = request.args.get("img_code_id")
    # 校验参数
    if not img_code_id:
        return abort(403)
    # 生成图片验证码
    img_name,img_code_text,img_code_bytes = captcha.generate_captcha()
    # 将图片和key保存到redis数据库中
    try:
        sr.set("img_code_id"+img_code_id,img_code_text ,ex=180)
    except BaseException as e:
        current_app.logger.error(e)
        return abort(500)
    # 返回验证码图片
    # 创建响应头
    response = make_response(img_code_bytes) # type:Response
    # 设置响应头
    response.content_type = "image/jpeg"

    return response


# 获取短信验证码
@passport_blu.route("/get_sms_code",methods=["POST"])
def get_sms_code():
    # 获取参数 request.json可以获取到application/json格式传过来的json数据
    img_code_id = request.json.get("img_code_id")
    img_code = request.json.get("img_code")
    mobile = request.json.get("mobile")
    # 校验参数
    if not all([img_code_id,img_code,mobile]):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 校验手机号格式
    if not re.match(r"^[1][3-8][0-9]{9}$",mobile):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 根据图片的key取出验证码文字
    try:
        real_img_code = sr.get("img_code_id"+img_code_id)
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg = error_map[RET.DBERR])
    # 校验图片验证码
    if not real_img_code: # 验证码是否已过期
        return jsonify(errno=RET.PARAMERR,errmsg = "验证码已过期")
    # print(img_code.upper())
    # print(real_img_code)
    if img_code.upper() != real_img_code:# 设置验证码忽略大小写 uppr将用户输入的转换为大写
        return jsonify(errno=RET.PARAMERR,errmsg = "验证码输入错误")
    # 从数据库中取出用户
    try:
        user_name = User.query.filter_by(mobile=mobile).first()
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR, errmsg=error_map[RET.DBERR])
    # 校验用户是否以存在
    if user_name:
        return jsonify(errno=RET.DATAEXIST, errmsg=error_map[RET.DATAEXIST])
    # 如果校验通过 发送短信
    # 生成四位随机数
    sms_code = "%04d" % random.randint(0,9999)
    current_app.logger.info("短信验证码为：%s" % sms_code)
    # 第三方集成的短信发送机制
    # res_code = CCP().send_template_sms(mobile,[sms_code,5],1)
    # if res_code == -1: # 短信发送失败
    #     return jsonify(errno=RET.THIRDERR,errmsg = error_map[RET.THIRDERR])
    # 将短信验证码保存到redis
    try:
        sr.set("sms_code_id"+mobile,sms_code,ex=60)
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg = error_map[RET.DBERR])
    # 将打印结果已json方式返回给前端
    return jsonify(errno=RET.OK,errmsg = error_map[RET.OK])


# 注册账号
@passport_blu.route("/register",methods=["POST"])
def register():
    # 获取参数 request.json可以获取到application/json格式传过来的json数据
    mobile = request.json.get("mobile")
    sms_code = request.json.get("sms_code")
    password = request.json.get("password")
    # 校验参数
    if not all([password,sms_code,mobile]):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 校验手机号格式
    if not re.match(r"^[1][3-8][0-9]{9}$",mobile):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 根据图片的key取出验证码文字
    try:
        real_sms_code = sr.get("sms_code_id"+mobile)
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg = error_map[RET.DBERR])

    # 校验图片验证码
    if not real_sms_code: # 验证码是否已过期
        return jsonify(errno=RET.PARAMERR,errmsg = "验证码已过期")

    if sms_code != real_sms_code:
        return jsonify(errno=RET.PARAMERR,errmsg = "验证码输入错误")
    # 将用户数据保存到数据库
    user = User()
    user.mobile = mobile
    # 设置计算属性对password进行加密处理(代码见 models.py @property)
    user.password = password
    user.nick_name = mobile
    # 记录用户最后登陆时间
    user.last_login = datetime.now()

    try:
        db.session.add(user)
        db.session.commit()
    except BaseException as e:
        current_app.logger.error(e)
        db.session.rollback()  # 设置回滚
        return jsonify(errno=RET.DBERR, errmsg=error_map[RET.DBERR])
    # 状态保持 免密登陆
    session["user_id"] = user.id
    # 将打印结果已json方式返回给前端
    return jsonify(errno=RET.OK,errmsg = error_map[RET.OK])


# 用户登录
@passport_blu.route("/login",methods=["POST"])
def login():
    # 获取参数 request.json可以获取到application/json格式传过来的json数据
    mobile = request.json.get("mobile")
    password = request.json.get("password")
    # 校验参数
    if not all([password,mobile]):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 校验手机号格式
    if not re.match(r"^[1][3-8][0-9]{9}$",mobile):
        return jsonify(errno=RET.PARAMERR,errmsg = error_map[RET.PARAMERR])
    # 根据手机号从数据库中取出用户模型
    try:
        user = User.query.filter_by(mobile=mobile).first()
    except BaseException as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg = error_map[RET.DBERR])

    # 判断该用户是否以存在
    if not user:
        return jsonify(errno=RET.USERERR,errmsg = error_map[RET.USERERR])
    # 校验用户密码
    if not user.check_password(password):
        return jsonify(errno=RET.PWDERR,errmsg = error_map[RET.PWDERR])

    # 设置最后登陆时间
    user.last_login = datetime.now()

    # 设置免密登陆（状态保持）
    session["user_id"] = user.id

    # 校验成功 给前段返回结果(已json格式返回)
    return jsonify(errno=RET.OK,errmsg = error_map[RET.OK])


# 退出登陆
@passport_blu.route("/logout")
def logout():
    # 将用户信息从session中删除 这里使用pop来删除 当键值对不存在时 不会报错 并且可以返回一个默认值
    session.pop("user_id",None)
    # 返回结果
    return jsonify(errno=RET.OK,errmsg = error_map[RET.OK])